Start of the pixbuf canvas item. Drawing affines seem to work. Bounding
authorFederico Mena Quintero <federico@redhat.com>
Tue, 2 Nov 1999 01:50:06 +0000 (01:50 +0000)
committerArturo Espinosa <unammx@src.gnome.org>
Tue, 2 Nov 1999 01:50:06 +0000 (01:50 +0000)
1999-11-01  Federico Mena Quintero  <federico@redhat.com>

* src/gnome-canvas-pixbuf.c: Start of the pixbuf canvas item.
Drawing affines seem to work.  Bounding box is fubared.  ::point()
is not implemented.  This is a work in progress.

* src/Makefile.am: Added gnome-canvas-pixbuf.[ch] to the list of
sources.

* configure.in: Sigh.  We need to link in libgnomeui for the
canvas stuff.  This mess will disappear when gdk-pixbuf gets
folded into gnome-libs.

gdk-pixbuf/ChangeLog
gdk-pixbuf/Makefile.am
gdk-pixbuf/gdk-pixbuf-io.h
gdk-pixbuf/gnome-canvas-pixbuf.c

index af84ad89680976c9e0a28c0e5e5b710a77828cff..7e8f6ba00ac6a08856008c386bb2c17b45700b28 100644 (file)
@@ -1,3 +1,16 @@
+1999-11-01  Federico Mena Quintero  <federico@redhat.com>
+
+       * src/gnome-canvas-pixbuf.c: Start of the pixbuf canvas item.
+       Drawing affines seem to work.  Bounding box is fubared.  ::point()
+       is not implemented.  This is a work in progress.
+
+       * src/Makefile.am: Added gnome-canvas-pixbuf.[ch] to the list of
+       sources.
+
+       * configure.in: Sigh.  We need to link in libgnomeui for the
+       canvas stuff.  This mess will disappear when gdk-pixbuf gets
+       folded into gnome-libs.
+
 1999-10-30    <jrb@redhat.com>
 
        * src/io-gif.c: Some more work.  Now it generates a gdk_pixbuf of
index 5db237704a9017dcddfe8426dd272293c80ca0ca..79a0eb4ee03c1765290767dc80423539431e995a 100644 (file)
@@ -37,7 +37,7 @@ AM_CPPFLAGS = "-DPIXBUF_LIBDIR=\"$(libexecdir)\""
 
 LDADDS = libgdk_pixbuf.la $(LIBART_LIBS) $(GLIB_LIBS)
 
-testpixbuf_LDADD = $(LDADDS) $(GTK_LIBS)
+testpixbuf_LDADD = $(LDADDS) $(LIBART_LIBS)
 
 #
 # The GdkPixBuf library
@@ -49,13 +49,15 @@ libgdk_pixbuf_la_SOURCES =  \
        gdk-pixbuf-data.c       \
        gdk-pixbuf-io.c         \
        gdk-pixbuf-loader.c     \
-       gdk-pixbuf-render.c
+       gdk-pixbuf-render.c     \
+       gnome-canvas-pixbuf.c
 
 libgdk_pixbuf_la_LDFLAGS = -version-info 1:0:0
 
 libgdk_pixbufinclude_HEADERS = \
        gdk-pixbuf.h            \
-       gdk-pixbuf-loader.h
+       gdk-pixbuf-loader.h     \
+       gnome-canvas-pixbuf.h
 
 noinst_HEADERS = \
        gdk-pixbuf-io.h
index 2fcd092573f55fe8f53e1dc55247e7f61d11e2e8..0e203b54ccf4309af056258a5ad9cdf87e269fb2 100644 (file)
  * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
  * Boston, MA 02111-1307, USA.
  */
+
+#ifndef GDK_PIXBUF_IO_H
+#define GDK_PIXBUF_IO_H
+
 #include <gmodule.h>
 #include "gdk-pixbuf.h"
 #include <stdio.h>
 
+#ifdef __cplusplus
+extern "C" {
+#pragma }
+#endif
+
 \f
 
 typedef void (* ModulePreparedNotifyFunc) (GdkPixbuf *pixbuf, gpointer user_data);
@@ -49,3 +58,10 @@ struct _GdkPixbufModule {
 GdkPixbufModule *gdk_pixbuf_get_module (gchar *buffer, gint size);
 void gdk_pixbuf_load_module (GdkPixbufModule *image_module);
 
+\f
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif
index ab1b1121601818d8ceffe538241c57c9f65f9608..abb05d46384dd83b0c5d23cadece08c665d1e46a 100644 (file)
@@ -22,7 +22,6 @@
 
 #include <config.h>
 #include <math.h>
-#include <libart_lgpl/art_rgb_affine_private.h>
 #include "gdk-pixbuf.h"
 #include "gnome-canvas-pixbuf.h"
 
@@ -213,7 +212,7 @@ gnome_canvas_pixbuf_set_arg (GtkObject *object, GtkArg *arg, guint arg_id)
                pixbuf = GTK_VALUE_POINTER (*arg);
                if (pixbuf != priv->pixbuf) {
                        if (pixbuf) {
-                               g_return_if_fail (pixbuf->art_pixbuf->format != ART_PIX_RGB);
+                               g_return_if_fail (pixbuf->art_pixbuf->format == ART_PIX_RGB);
                                g_return_if_fail (pixbuf->art_pixbuf->n_channels == 3
                                                  || pixbuf->art_pixbuf->n_channels == 4);
                                g_return_if_fail (pixbuf->art_pixbuf->bits_per_sample == 8);
@@ -327,40 +326,50 @@ gnome_canvas_pixbuf_get_arg (GtkObject *object, GtkArg *arg, guint arg_id)
 
 /* Bounds and utilities */
 
-/* Computes horizontal and vertical scaling vectors for the final transformation
- * to canvas pixel coordinates of a pixbuf canvas item.  The horizontal and
- * vertical dimensions may be specified in units or pixels, separately, so we
- * have to compute the components individually for each dimension.
+/* Computes the amount by which the unit horizontal and vertical vectors will be
+ * scaled by an affine transformation.
  */
 static void
-compute_xform_vectors (GnomeCanvasPixbuf *gcp, double *i2c, ArtPoint *i_c, ArtPoint *j_c)
+compute_xform_scaling (double *affine, ArtPoint *i_c, ArtPoint *j_c)
 {
-       PixbufPrivate *priv;
        ArtPoint orig, orig_c;
        ArtPoint i, j;
-       double length;
-
-       priv = gcp->priv;
 
-       /* Our origin */
+       /* Origin */
 
        orig.x = 0.0;
        orig.y = 0.0;
-       art_affine_point (&orig_c, &orig, i2c);
+       art_affine_point (&orig_c, &orig, affine);
 
        /* Horizontal and vertical vectors */
 
        i.x = 1.0;
        i.y = 0.0;
-       art_affine_point (&i_c, &i, i2c);
-       i_c.x -= orig_c.x;
-       i_c.y -= orig_c.y;
+       art_affine_point (i_c, &i, affine);
+       i_c->x -= orig_c.x;
+       i_c->y -= orig_c.y;
 
        j.x = 0.0;
        j.y = 1.0;
-       art_affine_point (&j_c, &j, i2c);
-       j_c.x -= orig_c.x;
-       j_c.y -= orig_c.y;
+       art_affine_point (j_c, &j, affine);
+       j_c->x -= orig_c.x;
+       j_c->y -= orig_c.y;
+}
+
+/* Computes horizontal and vertical scaling vectors for the final transformation
+ * to canvas pixel coordinates of a pixbuf canvas item.  The horizontal and
+ * vertical dimensions may be specified in units or pixels, separately, so we
+ * have to compute the components individually for each dimension.
+ */
+static void
+compute_xform_vectors (GnomeCanvasPixbuf *gcp, double *i2c, ArtPoint *i_c, ArtPoint *j_c)
+{
+       PixbufPrivate *priv;
+       double length;
+
+       priv = gcp->priv;
+
+       compute_xform_scaling (i2c, i_c, j_c);
 
        /* Compute the scaling vectors.  If a dimension is specified in pixels,
         * we normalize the base vector first so that it will represent the size
@@ -368,7 +377,7 @@ compute_xform_vectors (GnomeCanvasPixbuf *gcp, double *i2c, ArtPoint *i_c, ArtPo
         */
 
        if (priv->width_pixels) {
-               length = sqrt (i_c.x * i_c.x + i_c.y * i_c.y);
+               length = sqrt (i_c->x * i_c->x + i_c->y * i_c->y);
 
                if (length > GNOME_CANVAS_EPSILON) {
                        i_c->x /= length;
@@ -378,7 +387,7 @@ compute_xform_vectors (GnomeCanvasPixbuf *gcp, double *i2c, ArtPoint *i_c, ArtPo
        }
 
        if (priv->height_pixels) {
-               length = sqrt (j_c.x * j_c.x + j_c.y * j_c.y);
+               length = sqrt (j_c->x * j_c->x + j_c->y * j_c->y);
 
                if (length > GNOME_CANVAS_EPSILON) {
                        j_c->x /= length;
@@ -401,7 +410,6 @@ recompute_bounding_box (GnomeCanvasPixbuf *gcp)
        ArtPoint orig, orig_c;
        ArtPoint i_c, j_c;
        double w, h;
-       double length;
        double x1, y1, x2, y2, x3, y3, x4, y4;
        double mx1, my1, mx2, my2;
 
@@ -485,6 +493,9 @@ recompute_bounding_box (GnomeCanvasPixbuf *gcp)
                item->y1 = my2;
                item->y2 = my1;
        }
+
+       item->x2++;
+       item->y2++;
 }
 
 \f
@@ -509,7 +520,7 @@ gnome_canvas_pixbuf_update (GnomeCanvasItem *item, double *affine, ArtSVP *clip_
            || (flags & GNOME_CANVAS_UPDATE_AFFINE)
            || priv->need_pixbuf_update
            || priv->need_size_update)
-               gnome_canvas_request_redraw (item, item->x1, item->y1, item->x2, item->y2);
+               gnome_canvas_request_redraw (item->canvas, item->x1, item->y1, item->x2, item->y2);
 
        /* If we need a pixbuf update, or if the item changed visibility to
         * shown, recompute the bounding box.
@@ -517,26 +528,30 @@ gnome_canvas_pixbuf_update (GnomeCanvasItem *item, double *affine, ArtSVP *clip_
        if (priv->need_pixbuf_update
            || priv->need_size_update
            || ((flags & GNOME_CANVAS_UPDATE_VISIBILITY)
-               && (GTK_OBJECT_FLAGS (ii) & GNOME_CANVAS_ITEM_VISIBLE))
+               && (GTK_OBJECT_FLAGS (gcp) & GNOME_CANVAS_ITEM_VISIBLE))
            || (flags & GNOME_CANVAS_UPDATE_AFFINE)) {
                recompute_bounding_box (gcp);
                gnome_canvas_request_redraw (item->canvas, item->x1, item->y1, item->x2, item->y2);
                priv->need_pixbuf_update = FALSE;
                priv->need_size_update = FALSE;
        }
-       
+
 }
 
 \f
 
 /* Rendering */
 
+/* This is private to libart, but we need it.  Sigh. */
+extern void art_rgb_affine_run (int *p_x0, int *p_x1, int y, int src_width, int src_height,
+                               const double affine[6]);
+
 /* Fills the specified buffer with the transformed version of a pixbuf */
 static void
 transform_pixbuf (guchar *dest, int x, int y, int width, int height, int rowstride,
                  GdkPixbuf *pixbuf, double *affine)
 {
-       ArtPixbuf *apb;
+       ArtPixBuf *apb;
        int xx, yy;
        double inv[6];
        guchar *src, *d;
@@ -553,26 +568,26 @@ transform_pixbuf (guchar *dest, int x, int y, int width, int height, int rowstri
                dest_p.y = y + yy + 0.5;
 
                run_x1 = x;
-               run_x1 = x + width;
-               art_affine_run (&run_x1, &run_x2, yy + y,
-                               apb->width, apb->height,
-                               inv);
+               run_x2 = x + width;
+               art_rgb_affine_run (&run_x1, &run_x2, yy + y,
+                                   apb->width, apb->height,
+                                   inv);
 
                d = dest + yy * rowstride + (run_x1 - x) * 4;
 
-               for (xx = run_x0; xx < run-x1; xx++) {
-                       dest_p.x = x + 0.5;
+               for (xx = run_x1; xx < run_x2; xx++) {
+                       dest_p.x = xx + 0.5;
                        art_affine_point (&src_p, &dest_p, inv);
-                       src_x = floor (src_p.x + 0.5);
-                       src_y = floor (src_p.y + 0.5);
+                       src_x = floor (src_p.x);
+                       src_y = floor (src_p.y);
 
-                       src = apb->pixels + src_y * apb->rowstride + src_x * apb->num_channels;
+                       src = apb->pixels + src_y * apb->rowstride + src_x * apb->n_channels;
 
-                       for (i = 0; i < apb->num_channels; i++)
+                       for (i = 0; i < apb->n_channels; i++)
                                *d++ = *src++;
 
                        if (!apb->has_alpha)
-                               d++; /* It is already zeroed */
+                               *d++ = 255; /* opaque */
                }
        }
 }
@@ -588,6 +603,7 @@ gnome_canvas_pixbuf_draw (GnomeCanvasItem *item, GdkDrawable *drawable,
        ArtPoint i_c, j_c;
        double i_len, j_len;
        double scale[6], final[6];
+       double w, h;
        guchar *buf;
        GdkPixbuf *pixbuf;
 
@@ -600,13 +616,71 @@ gnome_canvas_pixbuf_draw (GnomeCanvasItem *item, GdkDrawable *drawable,
        /* Compute scaling factors and build the final affine */
 
        gnome_canvas_item_i2c_affine (item, i2c);
+       compute_xform_scaling (i2c, &i_c, &j_c);
+
+       i_len = sqrt (i_c.x * i_c.x + i_c.y * i_c.y);
+       j_len = sqrt (j_c.x * j_c.x + j_c.y * j_c.y);
+
+       if (priv->width_set)
+               w = priv->width;
+       else
+               w = priv->pixbuf->art_pixbuf->width;
+
+       if (priv->height_set)
+               h = priv->height;
+       else
+               h = priv->pixbuf->art_pixbuf->height;
+
+       /* Convert i_len and j_len into scaling factors */
+
+       if (priv->width_pixels) {
+               if (i_len > GNOME_CANVAS_EPSILON)
+                       i_len = 1.0 / i_len;
+               else
+                       i_len = 0.0;
+       } else
+               i_len = 1.0;
+
+       i_len *= w / priv->pixbuf->art_pixbuf->width;
+
+       if (priv->height_pixels) {
+               if (j_len > GNOME_CANVAS_EPSILON)
+                       j_len = 1.0 / j_len;
+               else
+                       j_len = 0.0;
+       } else
+               j_len = 1.0;
+
+       j_len *= h / priv->pixbuf->art_pixbuf->height;
+
+       /* Compute the final affine */
+
+       art_affine_scale (scale, i_len, j_len);
+       art_affine_multiply (final, scale, i2c);
+
+#if 0
        compute_xform_vectors (gcp, i2c, &i_c, &j_c);
 
        i_len = sqrt (i_c.x * i_c.x + i_c.y * i_c.y);
        j_len = sqrt (j_c.x * j_c.x + j_c.y * j_c.y);
 
+       if (priv->width_set)
+               i_len *= priv->width;
+       else
+               i_len *= priv->pixbuf->art_pixbuf->width;
+
+       i_len /= priv->pixbuf->art_pixbuf->width;
+
+       if (priv->height_set)
+               j_len *= priv->height;
+       else
+               j_len *= priv->pixbuf->art_pixbuf->height;
+
+       j_len /= priv->pixbuf->art_pixbuf->height;
+
        art_affine_scale (scale, i_len, j_len);
        art_affine_multiply (final, i2c, scale);
+#endif
 
        buf = g_new0 (guchar, width * height * 4);
        transform_pixbuf (buf, x, y, width, height, width * 4, priv->pixbuf, final);
@@ -641,5 +715,15 @@ gnome_canvas_pixbuf_point (GnomeCanvasItem *item, double x, double y, int cx, in
        gcp = GNOME_CANVAS_PIXBUF (item);
        priv = gcp->priv;
 
-       
+       *actual_item = item;
+
+       return 2 * item->canvas->close_enough;
+}
+
+\f
+
+/* Bounds handler for the pixbuf canvas item */
+static void
+gnome_canvas_pixbuf_bounds (GnomeCanvasItem *item, double *x1, double *y1, double *x2, double *y2)
+{
 }